<!DOCTYPE html>
<html>
<head><meta name="generator" content="Hexo 3.8.0">
  <meta charset="utf-8">

  <!-- PACE Progress Bar START -->
  
    <script src="https://raw.githubusercontent.com/HubSpot/pace/v1.0.2/pace.min.js"></script>
    <link rel="stylesheet" href="https://github.com/HubSpot/pace/raw/master/themes/orange/pace-theme-flash.css">
  
  

  <!-- PACE Progress Bar START -->

  
  <title>谈谈如何防止表单重复提交 | FunGa技术札记</title>
  <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
  
  <meta name="keywords" content="Web开发">
  
  
  
  
  <meta name="description" content="防止表单重复提交是我们在实际开发中经常需要考虑的事情，很多时候，我们可能在页面去做：当提交表单时，将按钮置为disabled，当获取到返回结果后将disabled去掉，这样保证当返回结果之前都无法再次发送请求。但是，很明显，只在前端做处理远远不够都，本文我们探讨都主要还是在后端做逻辑处理。">
<meta name="keywords" content="Web开发">
<meta property="og:type" content="article">
<meta property="og:title" content="谈谈如何防止表单重复提交">
<meta property="og:url" content="funga.cn/2019/01/11/form-submit-repeat/index.html">
<meta property="og:site_name" content="FunGa技术札记">
<meta property="og:description" content="防止表单重复提交是我们在实际开发中经常需要考虑的事情，很多时候，我们可能在页面去做：当提交表单时，将按钮置为disabled，当获取到返回结果后将disabled去掉，这样保证当返回结果之前都无法再次发送请求。但是，很明显，只在前端做处理远远不够都，本文我们探讨都主要还是在后端做逻辑处理。">
<meta property="og:locale" content="default">
<meta property="og:image" content="http://funga.oss-cn-beijing.aliyuncs.com/images/form-submit-1.png">
<meta property="og:updated_time" content="2019-01-13T06:53:32.097Z">
<meta name="twitter:card" content="summary">
<meta name="twitter:title" content="谈谈如何防止表单重复提交">
<meta name="twitter:description" content="防止表单重复提交是我们在实际开发中经常需要考虑的事情，很多时候，我们可能在页面去做：当提交表单时，将按钮置为disabled，当获取到返回结果后将disabled去掉，这样保证当返回结果之前都无法再次发送请求。但是，很明显，只在前端做处理远远不够都，本文我们探讨都主要还是在后端做逻辑处理。">
<meta name="twitter:image" content="http://funga.oss-cn-beijing.aliyuncs.com/images/form-submit-1.png">
  
    <link rel="alternate" href="/atom.xml" title="FunGa技术札记" type="application/atom+xml">
  
  <link rel="icon" href="/css/images/favicon.ico">
  
  <link href="https://fonts.googleapis.com/css?family=Open+Sans|Montserrat:700" rel="stylesheet" type="text/css">
  <link href="https://fonts.googleapis.com/css?family=Roboto:400,300,300italic,400italic" rel="stylesheet" type="text/css">
  <link href="https://cdn.bootcss.com/font-awesome/4.6.3/css/font-awesome.min.css" rel="stylesheet">
  <style type="text/css">
    @font-face{font-family:futura-pt;src:url(https://use.typekit.net/af/9749f0/00000000000000000001008f/27/l?subset_id=2&fvd=n5) format("woff2");font-weight:500;font-style:normal;}
    @font-face{font-family:futura-pt;src:url(https://use.typekit.net/af/90cf9f/000000000000000000010091/27/l?subset_id=2&fvd=n7) format("woff2");font-weight:500;font-style:normal;}
    @font-face{font-family:futura-pt;src:url(https://use.typekit.net/af/8a5494/000000000000000000013365/27/l?subset_id=2&fvd=n4) format("woff2");font-weight:lighter;font-style:normal;}
    @font-face{font-family:futura-pt;src:url(https://use.typekit.net/af/d337d8/000000000000000000010095/27/l?subset_id=2&fvd=i4) format("woff2");font-weight:400;font-style:italic;}</style>
    
  <link rel="stylesheet" id="athemes-headings-fonts-css" href="//fonts.googleapis.com/css?family=Yanone+Kaffeesatz%3A200%2C300%2C400%2C700&amp;ver=4.6.1" type="text/css" media="all">
  <link rel="stylesheet" href="/css/style.css">

  <script src="https://code.jquery.com/jquery-3.1.1.min.js"></script>

  <!-- Bootstrap core CSS -->
  <link rel="stylesheet" href="https://netdna.bootstrapcdn.com/bootstrap/3.0.2/css/bootstrap.min.css">
  <link rel="stylesheet" href="/css/hiero.css">
  <link rel="stylesheet" href="/css/glyphs.css">
  
    <link rel="stylesheet" href="/css/vdonate.css">
  

  <!-- Custom CSS -->
  <link rel="stylesheet" href="/css/my.css">
  <!-- Google Adsense -->
  
<link rel="stylesheet" href="/css/prism-tomorrow.css" type="text/css"></head>
</html>
<script>
var themeMenus = {};

  themeMenus["/"] = "Home"; 

  themeMenus["/archives"] = "Archives"; 

  themeMenus["/categories"] = "Categories"; 

  themeMenus["/tags"] = "Tags"; 

  themeMenus["/about"] = "About"; 

</script>


  <body data-spy="scroll" data-target="#toc" data-offset="50">


  <header id="allheader" class="site-header" role="banner">
  <div class="clearfix container">
      <div class="site-branding">

          <h1 class="site-title">
            
              <a href="/" title="FunGa技术札记" rel="home"> FunGa技术札记 </a>
            
          </h1>

          
            
          <nav id="main-navigation" class="main-navigation" role="navigation">
            <a class="nav-open">Menu</a>
            <a class="nav-close">Close</a>
            <div class="clearfix sf-menu">

              <ul id="main-nav" class="nmenu sf-js-enabled">
                    
                      <li class="menu-item menu-item-type-custom menu-item-object-custom menu-item-home menu-item-1663"> <a class="" href="/">Home</a> </li>
                    
                      <li class="menu-item menu-item-type-custom menu-item-object-custom menu-item-home menu-item-1663"> <a class="" href="/archives">Archives</a> </li>
                    
                      <li class="menu-item menu-item-type-custom menu-item-object-custom menu-item-home menu-item-1663"> <a class="" href="/categories">Categories</a> </li>
                    
                      <li class="menu-item menu-item-type-custom menu-item-object-custom menu-item-home menu-item-1663"> <a class="" href="/tags">Tags</a> </li>
                    
                      <li class="menu-item menu-item-type-custom menu-item-object-custom menu-item-home menu-item-1663"> <a class="" href="/about">About</a> </li>
                    
              </ul>
            </div>
          </nav>


      </div>
  </div>
</header>




  <div id="container">
    <div id="wrap">
            
      <div id="content" class="outer">
        
          <section id="main" style="float:none;"><article id="post-form-submit-repeat" style="width: 66%; float:left;" class="article article-type-post" itemscope="" itemprop="blogPost">
  <div id="articleInner" class="clearfix post-1016 post type-post status-publish format-standard has-post-thumbnail hentry category-template-2 category-uncategorized tag-codex tag-edge-case tag-featured-image tag-image tag-template">
    
    
      <header class="article-header">
        
  
    <h1 class="thumb" itemprop="name">
      谈谈如何防止表单重复提交
    </h1>
  

      </header>
    
    <div class="article-meta">
      
	Posted on <a href="/2019/01/11/form-submit-repeat/" class="article-date">
	  <time datetime="2019-01-11T04:39:17.000Z" itemprop="datePublished">January 11, 2019</time>
	</a>

      
	<span id="busuanzi_container_page_pv">
	  本文总阅读量<span id="busuanzi_value_page_pv"></span>次
	</span>

    </div>
    <div class="article-entry" itemprop="articleBody">
      
        <p>防止表单重复提交是我们在实际开发中经常需要考虑的事情，很多时候，我们可能在页面去做：当提交表单时，将按钮置为disabled，当获取到返回结果后将disabled去掉，这样保证当返回结果之前都无法再次发送请求。但是，很明显，只在前端做处理远远不够都，本文我们探讨都主要还是在后端做逻辑处理。</p>
<a id="more"></a>
<h3 id="场景一-简单都防止手抖双击（连击）"><a href="#场景一-简单都防止手抖双击（连击）" class="headerlink" title="场景一 简单都防止手抖双击（连击）"></a>场景一 简单都防止手抖双击（连击）</h3><p>用户’手抖’是常见不过的事情，手快也好，鼠标敏感也好，着急提交连续点击鼠标也罢，总的来说，这样的情况下，用户就会出现多次提交表单。造成的现象就是：假设提交报销申请，那么就会出现两单（或多单）内容完全一样的报销申请，很明显，这是不对的。</p>
<p>这里又分两种情况：</p>
<h4 id="1、第一次请求结果还没有返回又发起第二次请求"><a href="#1、第一次请求结果还没有返回又发起第二次请求" class="headerlink" title="1、第一次请求结果还没有返回又发起第二次请求"></a>1、第一次请求结果还没有返回又发起第二次请求</h4><p>前面介绍了，这种情况也可在前端处理，当然，我们在后端也得做逻辑判断。这种情况我们也是必须要处理的，如果网络或者接口比较慢，存在连击的话，那么将会产生非常多的相同的数据。</p>
<p>解决方案：<br>在提交表单前，先获取一个token，这个token在后端生成，比如可以根据jsessionId、登陆用户等可以定位当前用户以及请求等客户端的数据来生成。</p>
<pre class=" language-java"><code class="language-java">String token <span class="token operator">=</span> <span class="token function">MD5</span><span class="token punctuation">(</span>sessionId<span class="token operator">+</span>userId<span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre>
<p>这个token相当于一个标记，或者类似一把锁，拿到token的才能发送表单提交请求，没有拿到的都不发。</p>
<ul>
<li>前端发送Ajax请求，获取token。先从Redis取token，如果存在，则表示这个客户端有一个请求还没有返回，则不给token，返回错误code。如果不存在token，则生成token返回。</li>
<li>前端如果获取到了错误码，表示之前有请求在处理，则不继续发送请求（根据情况可以决定是否进行提示）；如果获取到了token，则表示可以进行表单到提交。后台拿到token，先校验Redis中是否存在，如果存在，则进行处理，比如保存报销单，逻辑处理完成后将token从Redis中移除。当然，如果不存在token，说明此次请求是不处理的。（使用aop实现完美）</li>
</ul>
<p><img src="http://funga.oss-cn-beijing.aliyuncs.com/images/form-submit-1.png" alt=""></p>
<h4 id="2、第一次请求已经结束才发起的第二次请求"><a href="#2、第一次请求已经结束才发起的第二次请求" class="headerlink" title="2、第一次请求已经结束才发起的第二次请求"></a>2、第一次请求已经结束才发起的第二次请求</h4><p>这种情况与前面一种其实不一样，当前两次请求其实是相互独立的，我提交了一次报销申请，我肯定还可以再次提交一次报销申请，如果我们做了上述的防止表单重复提交校验，那么这种情况下又需要做什么呢？</p>
<p>后端处理到前端获取到结果可能是存在网络延迟的，即使结果已经返回了，基于原来表单的数据，可能存在连击的情况，也就是说，可能将完全一模一样的表单数据再次进行了提交，并且不在第一次情况内，如果是报销单，那么我依然可能被保存了两条一模一样的报销申请。</p>
<p>解决方案：</p>
<p>我们以报销为例，如果短时间内提交相同内容的报销内容，我们是否可以认为是重复提交呢？对于报销来说，相同的报销，我们可以把具有相同报销金额、报销类型、报销说明等字段相同则认为相同（根据实际情况）。</p>
<ul>
<li>前端提交表单，我们根据关键字段的内容生成一个token（拥有相同的内容，token必然相同），我们去Redis中查询是否有此token：如果有，则表示表单算重复提交，本次请求不处理或者返回上次的结果；如果没有，则将token存于Redis。</li>
<li>我们保存报销数据，处理完成后，我们将结果返回之前也存入Redis，这个和之前token一样，需要设置一个失效时间（比如5秒失效，5秒内不可以发送同样数据的表单）</li>
</ul>
<p>这样，我们可以保证5秒内，不可能提交完全数据一样的表单并且都会被处理。</p>
<p>由于我们系统一般集群部署，所以需要使用到Redis。</p>

      
    </div>
    <footer class="entry-meta entry-footer">
      
	<span class="ico-folder"></span>
    <a class="article-category-link" href="/categories/其它分类/">其它分类</a>

      
  <span class="ico-tags"></span>
  <ul class="article-tag-list"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/Web开发/">Web开发</a></li></ul>

      
        <div id="donation_div"></div>

<script src="/js/vdonate.js"></script>
<script>
var a = new Donate({
  title: '如果觉得我的文章对您有用，请随意打赏。您的支持将鼓励我继续创作!', // 可选参数，打赏标题
  btnText: 'Donate', // 可选参数，打赏按钮文字
  el: document.getElementById('donation_div'),
  wechatImage: 'https://funga.oss-cn-beijing.aliyuncs.com/config/wechat_pay_code.jpg',
  alipayImage: 'https://funga.oss-cn-beijing.aliyuncs.com/config/alipay_code.png'
});
</script>
      
            
      
        <!--PC版-->
<div id="SOHUCS"></div>
<!-- <script charset="utf-8" type="text/javascript" src="https://changyan.sohu.com/upload/changyan.js" ></script> -->
<script charset="utf-8" type="text/javascript" src="https://funga.oss-cn-beijing.aliyuncs.com/static/changyan.js"></script>
<script type="text/javascript">
window.changyan.api.config({
appid: 'cyrV0wIGI',
conf: 'prod_3d29a99dcddb3782e98f1fa951b9868f'
});
</script>
      
    </footer>
  </div>
  
    
<nav id="article-nav">
  
  
    <a href="/2018/10/10/sp-e/" id="article-nav-older" class="article-nav-link-wrap">
      <strong class="article-nav-caption">Older</strong>
      <div class="article-nav-title">Java Web项目全局异常捕获实现</div>
    </a>
  
</nav>

  
</article>

<!-- Table of Contents -->

  <aside id="sidebar">
    <div id="toc" class="toc-article" style="overflow-y: scroll; max-width: 28%;">
    <strong class="toc-title">Contents</strong>
    
      <ol class="nav"><li class="nav-item nav-level-3"><a class="nav-link" href="#场景一-简单都防止手抖双击（连击）"><span class="nav-number">1.</span> <span class="nav-text">场景一 简单都防止手抖双击（连击）</span></a><ol class="nav-child"><li class="nav-item nav-level-4"><a class="nav-link" href="#1、第一次请求结果还没有返回又发起第二次请求"><span class="nav-number">1.1.</span> <span class="nav-text">1、第一次请求结果还没有返回又发起第二次请求</span></a></li><li class="nav-item nav-level-4"><a class="nav-link" href="#2、第一次请求已经结束才发起的第二次请求"><span class="nav-number">1.2.</span> <span class="nav-text">2、第一次请求已经结束才发起的第二次请求</span></a></li></ol></li></ol>
    
    </div>
  </aside>
</section>
        
      </div>
      <footer id="footer" class="site-footer">
  

  <div class="clearfix container">
      <div class="site-info">
	      &copy; 2019 FunGa技术札记 All Rights Reserved.
          
            <span id="busuanzi_container_site_uv">
              本站访客数<span id="busuanzi_value_site_uv"></span>人次  
              本站总访问量<span id="busuanzi_value_site_pv"></span>次
            </span>
          
      </div>
      <div class="site-credit">
        Theme by <a href="https://github.com/iTimeTraveler/hexo-theme-hiero" target="_blank">hiero</a>
      </div>
  </div>
</footer>


<!-- min height -->

<script>
    var contentdiv = document.getElementById("content");

    contentdiv.style.minHeight = document.body.offsetHeight - document.getElementById("allheader").offsetHeight - document.getElementById("footer").offsetHeight + "px";
</script>

<!-- Custome JS -->
<script src="/js/my.js"></script>
    </div>
    <!-- <nav id="mobile-nav">
  
    <a href="/" class="mobile-nav-link">Home</a>
  
    <a href="/archives" class="mobile-nav-link">Archives</a>
  
    <a href="/categories" class="mobile-nav-link">Categories</a>
  
    <a href="/tags" class="mobile-nav-link">Tags</a>
  
    <a href="/about" class="mobile-nav-link">About</a>
  
</nav> -->
    

<!-- mathjax config similar to math.stackexchange -->

<script type="text/x-mathjax-config">
  MathJax.Hub.Config({
    tex2jax: {
      inlineMath: [ ['$','$'], ["\\(","\\)"] ],
      processEscapes: true
    }
  });
</script>

<script type="text/x-mathjax-config">
    MathJax.Hub.Config({
      tex2jax: {
        skipTags: ['script', 'noscript', 'style', 'textarea', 'pre', 'code']
      }
    });
</script>

<script type="text/x-mathjax-config">
    MathJax.Hub.Queue(function() {
        var all = MathJax.Hub.getAllJax(), i;
        for(i=0; i < all.length; i += 1) {
            all[i].SourceElement().parentNode.className += ' has-jax';
        }
    });
</script>

<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.4/MathJax.js?config=TeX-AMS-MML_HTMLorMML">
</script>



  <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fancybox/2.1.5/jquery.fancybox.min.css">
  <script src="https://cdnjs.cloudflare.com/ajax/libs/fancybox/2.1.5/jquery.fancybox.min.js"></script>


<script src="/js/scripts.js"></script>
<script src="https://stackpath.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js"></script>
<script src="/js/main.js"></script>







  <div style="display: none;">
    <script src="https://s95.cnzz.com/z_stat.php?id=1260716016&web_id=1260716016" language="JavaScript"></script>
  </div>



	<script async src="https://dnqof95d40fo6.cloudfront.net/atw7f8.js">
	</script>






  </div>

  <a id="rocket" href="#top" class=""></a>
  <script type="text/javascript" src="/js/totop.js" async=""></script>
</body>
</html>
